package main

import (
	"Go_Advanced/bluebell/controller"
	"Go_Advanced/bluebell/dao/mysql"
	"Go_Advanced/bluebell/dao/redis"
	"Go_Advanced/bluebell/logger"
	"Go_Advanced/bluebell/pkg/snowflake/snowflake"
	"Go_Advanced/bluebell/router"
	"Go_Advanced/bluebell/setting"
	"fmt"
	_ "github.com/go-sql-driver/mysql"
	_ "github.com/jinzhu/gorm/dialects/mysql"
	"go.uber.org/zap"
)

//Go Web开发较通用的脚手架模板

func main() {
	//1.加载配置
	if err := setting.Init(); err != nil {
		fmt.Printf("init setting failed,err:%v\n", err)
		return
	}
	//2.初始化日志
	if err := logger.Init(setting.Conf.LogConfig, setting.Conf.AppConfig.Mode); err != nil {
		fmt.Printf("init logger failed,err:%v\n", err)
		return
	}
	defer zap.L().Sync()
	zap.L().Debug("logger init success ...")
	//3.初始化mysql连接
	if err := mysql.Init(setting.Conf.MySQLConfig); err != nil {
		fmt.Printf("init mysql failed,err:%v\n", err)
		return
	}
	defer mysql.Close()
	//4.初始化Redis连接
	if err := redis.Init(setting.Conf.RedisConfig); err != nil {
		fmt.Printf("init redis failed,err:%v\n", err)
		return
	}
	//defer redis.Close()
	//初始化雪花算法
	if err := snowflake.Init(setting.Conf.StartTime, setting.Conf.MachineID); err != nil {
		fmt.Printf("init snowflake failed,err:%v\n", err)
		return
	}
	//初始化gin框架内置的校验器使用的翻译器
	if err := controller.InitTrans("zh"); err != nil {
		fmt.Printf("init validator trans failed,err:%v\n", err)
		return
	}
	//5.注册路由
	r := router.Setup(setting.Conf.AppConfig.Mode)
	err := r.Run(fmt.Sprintf(":%d", setting.Conf.AppConfig.Port))
	//r.Run()
	if err != nil {
		fmt.Printf("run server failed,err:%v\n", err)
	}

	//6。启动服务（优雅关机
	//srv := &http.Server{
	//	Addr:    fmt.Sprintf("%d", viper.GetInt("app.port")),
	//	Handler: r,
	//}
	//
	//go func() {
	//	// 开启一个goroutine启动服务
	//	if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
	//		log.Fatalf("listen: %s\n", err)
	//	}
	//}()
	//
	//// 等待中断信号来优雅地关闭服务器，为关闭服务器操作设置一个5秒的超时
	//quit := make(chan os.Signal, 1) // 创建一个接收信号的通道
	//// kill 默认会发送 syscall.SIGTERM 信号
	//// kill -2 发送 syscall.SIGINT 信号，我们常用的Ctrl+C就是触发系统SIGINT信号
	//// kill -9 发送 syscall.SIGKILL 信号，但是不能被捕获，所以不需要添加它
	//// signal.Notify把收到的 syscall.SIGINT或syscall.SIGTERM 信号转发给quit
	//signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) // 此处不会阻塞
	//<-quit                                               // 阻塞在此，当接收到上述两种信号时才会往下执行
	//zap.L().Info("Shutdown Server ...")
	//// 创建一个5秒超时的context
	//ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
	//defer cancel()
	//// 5秒内优雅关闭服务（将未处理完的请求处理完再关闭服务），超过5秒就超时退出
	//if err := srv.Shutdown(ctx); err != nil {
	//	zap.L().Fatal("Server Shutdown: ", zap.Error(err))
	//}
	//
	//zap.L().Info("Server exiting")
}
